<!DOCTYPE html>
<html>
<head>
<title>Redux</title>
<meta http-equiv="Content-Type" content="text/html; charset=utf-8" />
<style type="text/css">
html,
body,
h1,
h2,
h3,
h4,
h5,
h6,
p,
blockquote,
ol,
ul,
li,
img {
    margin: 0;
    padding: 0;
    font-size: 100%;
    font: inherit;
    vertical-align: baseline;
}

html * {
    font-family: "ff-din-web-pro-1", "ff-din-web-pro-2", sans-serif;
    font-size: 16px;
    line-height: 19.2px;
    color-profile: sRGB;
}

body {
    min-width: 32em;
    max-width: 56em;
    margin: 10px auto;
}

p, blockquote p {
    line-height: 1.6;
}

ul, ol {
    margin: 16px 0;
}

ul li, ol li {
    line-height: 1.6;
}

p {
    font-weight: lighter;
    margin: 10px 0;
}

strong {
    font-weight: bold;
}

ol,
ul {
    margin-left: 2em;
}

h1,
h2,
h3,
h4,
h5,
h6 {
    font-weight: lighter;
    text-transform: capitalize;
    margin: 20px 0;
    border-bottom: 1px solid;
    padding-bottom: 6px;
}

h1, h1 > code {
    font-size: 24.624px;
    line-height: 29.548799999999996px;
}

h2, h2 > code {
    font-size: 24.624px;
    line-height: 29.548799999999996px;
}

h3, h3 > code {
    font-size: 23.44px;
    line-height: 28.128px;
}

h4, h4 > code {
    font-size: 22.16px;
    line-height: 26.592px;
}

h5, h5 > code {
    font-size: 22.16px;
    line-height: 26.592px;
}

h6, h6 > code {
    font-size: 22.16px;
    line-height: 26.592px;
}

img {
    margin-bottom: 20px;
}

h1 img,
h2 img,
h3 img,
h4 img,
h5 img,
h6 img,
p img {
    margin-bottom: 0;
}

pre,
code {
    font-family: monospace, Consolas, "Source Code Pro", Arial, sans-serif;
    color: #586e75;
    background-color: #eee8d5;
}

pre {
    white-space: pre-wrap;
    word-wrap: break-word;
    padding: 12px;
    margin-bottom: 20px;
}

code {
    border-radius: 3px;
}

h1 {
    text-transform: uppercase;
    font-weight: bold;
}

h3,
h4,
h5,
h6 {
    border-bottom: none;
}

html body {
    background-color: #fdf6e3;
}

html h1,
html h2,
html h3,
html h4,
html h5,
html h6 {
    color: #586e75;
    border-color: #657b83;
}

html a,
html a:active,
html a:visited {
    color: #586e75;
    text-decoration: none;
    border-bottom: 1px dashed;
    border-radius: 2px;
}

html a:hover {
    background-color: #eee8d5;
}

blockquote a:hover {
    background-color: #fdf6e3;
}

html a,
html a:active,
html a:visited,
html code.url {
    color: #b58900;
}

html h1 {
    color: #b58900;
}

html h2,
html h3,
html h4,
html h5,
html h6 {
    color: #b58900;
}

/* QUOTES
=============================================================================*/
blockquote {
    border-left: 4px solid #b58900;
    padding: 12px;
    background: #eee8d5;
    border-bottom-right-radius: 2px;
}

blockquote code {
    background: #fdf6e3;
}

blockquote > :first-child {
    margin-top: 0;
}

blockquote > :last-child {
    margin-bottom: 0;
}

/* TABLES
=============================================================================*/
table {
    margin: 0 auto;
    border-collapse: collapse;
    width: 100%;
    box-sizing: border-box;
    margin-bottom: 30px;
}

table th, table td {
    border: 1px solid #ccc;
    padding: 6px 13px;
}

table td {
    word-break: break-word;
    line-height: 1.3;
}

table th {
    font-weight: bold;
    text-align: center !important;
    background-color: #eee8d5;
}

table tr {
    border-top: 1px solid #ccc;
    background-color: #fdf6e3;
}

/* IMAGES
=============================================================================*/
img {
    max-width: 100%;
}

p > img {
    display: table;
    margin: 0 auto;
}

p code, li code, td code {
    padding: 1px 3px;
    border-radius: 3px;
}

.cp_embed_wrapper {
    margin: 20px 0;
}

.hljs {
	background: #eee8d5 !important;
}

@media screen and (min-width: 980px) and (max-width: 980px) {	
    table thead tr th,
    table thead tr th > code,
    table tbody tr td,
    table tbody tr td > code,
    table tbody tr td > strong {
        font-size: 1.3em;
        line-height: 1.3;
    }

    p, p code,
    p strong, p strong > code,
    blockquote p {
        font-size: 1.3em;
        line-height: 1.6;
    }

    pre > code,
    ul li pre > code,
    ol li pre > code{
		font-size: 1.3em;
        line-height: 1.3;    	
    }	
	

    ul li, ol li,
    ul li > code,
    ol li > code {
        font-size: 1.3em;
		line-height: 1.3;          
    }

    ul {
        margin-left: 3.4em;
    }

    ol {
        margin-left: 3.6em;
    }
}
</style>
<style type="text/css">
.highlight  { background: #ffffff; }
.highlight .c { color: #999988; font-style: italic } /* Comment */
.highlight .err { color: #a61717; background-color: #e3d2d2 } /* Error */
.highlight .k { font-weight: bold } /* Keyword */
.highlight .o { font-weight: bold } /* Operator */
.highlight .cm { color: #999988; font-style: italic } /* Comment.Multiline */
.highlight .cp { color: #999999; font-weight: bold } /* Comment.Preproc */
.highlight .c1 { color: #999988; font-style: italic } /* Comment.Single */
.highlight .cs { color: #999999; font-weight: bold; font-style: italic } /* Comment.Special */
.highlight .gd { color: #000000; background-color: #ffdddd } /* Generic.Deleted */
.highlight .gd .x { color: #000000; background-color: #ffaaaa } /* Generic.Deleted.Specific */
.highlight .ge { font-style: italic } /* Generic.Emph */
.highlight .gr { color: #aa0000 } /* Generic.Error */
.highlight .gh { color: #999999 } /* Generic.Heading */
.highlight .gi { color: #000000; background-color: #ddffdd } /* Generic.Inserted */
.highlight .gi .x { color: #000000; background-color: #aaffaa } /* Generic.Inserted.Specific */
.highlight .go { color: #888888 } /* Generic.Output */
.highlight .gp { color: #555555 } /* Generic.Prompt */
.highlight .gs { font-weight: bold } /* Generic.Strong */
.highlight .gu { color: #aaaaaa } /* Generic.Subheading */
.highlight .gt { color: #aa0000 } /* Generic.Traceback */
.highlight .kc { font-weight: bold } /* Keyword.Constant */
.highlight .kd { font-weight: bold } /* Keyword.Declaration */
.highlight .kp { font-weight: bold } /* Keyword.Pseudo */
.highlight .kr { font-weight: bold } /* Keyword.Reserved */
.highlight .kt { color: #445588; font-weight: bold } /* Keyword.Type */
.highlight .m { color: #009999 } /* Literal.Number */
.highlight .s { color: #d14 } /* Literal.String */
.highlight .na { color: #008080 } /* Name.Attribute */
.highlight .nb { color: #0086B3 } /* Name.Builtin */
.highlight .nc { color: #445588; font-weight: bold } /* Name.Class */
.highlight .no { color: #008080 } /* Name.Constant */
.highlight .ni { color: #800080 } /* Name.Entity */
.highlight .ne { color: #990000; font-weight: bold } /* Name.Exception */
.highlight .nf { color: #990000; font-weight: bold } /* Name.Function */
.highlight .nn { color: #555555 } /* Name.Namespace */
.highlight .nt { color: #000080 } /* Name.Tag */
.highlight .nv { color: #008080 } /* Name.Variable */
.highlight .ow { font-weight: bold } /* Operator.Word */
.highlight .w { color: #bbbbbb } /* Text.Whitespace */
.highlight .mf { color: #009999 } /* Literal.Number.Float */
.highlight .mh { color: #009999 } /* Literal.Number.Hex */
.highlight .mi { color: #009999 } /* Literal.Number.Integer */
.highlight .mo { color: #009999 } /* Literal.Number.Oct */
.highlight .sb { color: #d14 } /* Literal.String.Backtick */
.highlight .sc { color: #d14 } /* Literal.String.Char */
.highlight .sd { color: #d14 } /* Literal.String.Doc */
.highlight .s2 { color: #d14 } /* Literal.String.Double */
.highlight .se { color: #d14 } /* Literal.String.Escape */
.highlight .sh { color: #d14 } /* Literal.String.Heredoc */
.highlight .si { color: #d14 } /* Literal.String.Interpol */
.highlight .sx { color: #d14 } /* Literal.String.Other */
.highlight .sr { color: #009926 } /* Literal.String.Regex */
.highlight .s1 { color: #d14 } /* Literal.String.Single */
.highlight .ss { color: #990073 } /* Literal.String.Symbol */
.highlight .bp { color: #999999 } /* Name.Builtin.Pseudo */
.highlight .vc { color: #008080 } /* Name.Variable.Class */
.highlight .vg { color: #008080 } /* Name.Variable.Global */
.highlight .vi { color: #008080 } /* Name.Variable.Instance */
.highlight .il { color: #009999 } /* Literal.Number.Integer.Long */
.pl-c {
    color: #969896;
}

.pl-c1,.pl-mdh,.pl-mm,.pl-mp,.pl-mr,.pl-s1 .pl-v,.pl-s3,.pl-sc,.pl-sv {
    color: #0086b3;
}

.pl-e,.pl-en {
    color: #795da3;
}

.pl-s1 .pl-s2,.pl-smi,.pl-smp,.pl-stj,.pl-vo,.pl-vpf {
    color: #333;
}

.pl-ent {
    color: #63a35c;
}

.pl-k,.pl-s,.pl-st {
    color: #a71d5d;
}

.pl-pds,.pl-s1,.pl-s1 .pl-pse .pl-s2,.pl-sr,.pl-sr .pl-cce,.pl-sr .pl-sra,.pl-sr .pl-sre,.pl-src,.pl-v {
    color: #df5000;
}

.pl-id {
    color: #b52a1d;
}

.pl-ii {
    background-color: #b52a1d;
    color: #f8f8f8;
}

.pl-sr .pl-cce {
    color: #63a35c;
    font-weight: bold;
}

.pl-ml {
    color: #693a17;
}

.pl-mh,.pl-mh .pl-en,.pl-ms {
    color: #1d3e81;
    font-weight: bold;
}

.pl-mq {
    color: #008080;
}

.pl-mi {
    color: #333;
    font-style: italic;
}

.pl-mb {
    color: #333;
    font-weight: bold;
}

.pl-md,.pl-mdhf {
    background-color: #ffecec;
    color: #bd2c00;
}

.pl-mdht,.pl-mi1 {
    background-color: #eaffea;
    color: #55a532;
}

.pl-mdr {
    color: #795da3;
    font-weight: bold;
}

.pl-mo {
    color: #1d3e81;
}
.task-list {
padding-left:10px;
margin-bottom:0;
}

.task-list li {
    margin-left: 20px;
}

.task-list-item {
list-style-type:none;
padding-left:10px;
}

.task-list-item label {
font-weight:400;
}

.task-list-item.enabled label {
cursor:pointer;
}

.task-list-item+.task-list-item {
margin-top:3px;
}

.task-list-item-checkbox {
display:inline-block;
margin-left:-20px;
margin-right:3px;
vertical-align:1px;
}
</style>
<base target=_blank>
<meta http-equiv="X-UA-Compatible" content="IE=edge,Chrome=1">
<meta name="keywords" content="whjin,前端开发文档,html,css,javascript,canvas,jquery,vue.js,http,ajax,git,webpack">
<meta name="format-detection" content="telephone=no">
<meta name="description" content="前端开发文档">
<meta name="author" content="whjin">
<link rel="shortcut icon" href="https://whjin.github.io/frontend-dev-doc/images/logo.png">
<a href="https://github.com/whjin" class="github-corner" aria-label="View source on GitHub" target="_blank"><svg width="80" height="80" viewBox="0 0 250 250" style="fill:#151513; color:#fff; position: fixed; top: 0; border: 0; right: 0;" aria-hidden="true"><path d="M0,0 L115,115 L130,115 L142,142 L250,250 L250,0 Z"></path><path d="M128.3,109.0 C113.8,99.7 119.0,89.6 119.0,89.6 C122.0,82.7 120.5,78.6 120.5,78.6 C119.2,72.0 123.4,76.3 123.4,76.3 C127.3,80.9 125.5,87.3 125.5,87.3 C122.9,97.6 130.6,101.9 134.4,103.2" fill="currentColor" style="transform-origin: 130px 106px;" class="octo-arm"></path><path d="M115.0,115.0 C114.9,115.1 118.7,116.5 119.8,115.4 L133.7,101.6 C136.9,99.2 139.9,98.4 142.2,98.6 C133.8,88.0 127.5,74.4 143.8,58.0 C148.5,53.4 154.0,51.2 159.7,51.0 C160.3,49.4 163.2,43.6 171.4,40.1 C171.4,40.1 176.1,42.5 178.8,56.2 C183.1,58.6 187.2,61.8 190.9,65.4 C194.5,69.0 197.7,73.2 200.1,77.6 C213.8,80.2 216.3,84.9 216.3,84.9 C212.7,93.1 206.9,96.0 205.4,96.6 C205.1,102.4 203.0,107.8 198.3,112.5 C181.9,128.9 168.3,122.5 157.7,114.1 C157.9,116.9 156.7,120.9 152.7,124.9 L141.0,136.5 C139.8,137.7 141.6,141.9 141.8,141.8 Z" fill="currentColor" class="octo-body"></path></svg></a><style>.github-corner:hover .octo-arm{animation:octocat-wave 560ms ease-in-out}@keyframes octocat-wave{0%,100%{transform:rotate(0)}20%,60%{transform:rotate(-25deg)}40%,80%{transform:rotate(10deg)}}@media (max-width:500px){.github-corner:hover .octo-arm{animation:none}.github-corner .octo-arm{animation:octocat-wave 560ms ease-in-out}}</style>
<link href="https://cdn.bootcss.com/highlight.js/9.15.6/styles/a11y-light.min.css" rel="stylesheet">
<script src="https://cdn.bootcss.com/highlight.js/9.15.6/highlight.min.js"></script>
<script >hljs.initHighlightingOnLoad();</script> 
</head>
<body>
<h1 id="redux">Redux</h1>
<h2 id="redux-">Redux-修改共享状态</h2>
<p><code>Redux</code> 是一种架构模式（<code>Flux</code> 架构的一种变种），它不关注你到底用什么库，你可以把它应用到 <code>React</code> 和 <code>Vue</code>，甚至跟 <code>jQuery</code> 结合都没有问题。而 <code>React-redux</code> 就是把 <code>Redux</code> 这种架构模式和 <code>React.js</code> 结合起来的一个库，就是 <code>Redux</code> 架构在 <code>React.js</code> 中的体现。</p>
<p><p class="codepen" data-height="265" data-theme-id="0" data-default-tab="js" data-user="whjin" data-slug-hash="EroKVZ" style="height: 265px; box-sizing: border-box; display: flex; align-items: center; justify-content: center; border: 2px solid black; margin: 1em 0; padding: 1em;" data-pen-title="修改共享状态"><br>  <span>See the Pen <a href="https://codepen.io/whjin/pen/EroKVZ/"><br>  修改共享状态</a> by whjin (<a href="https://codepen.io/whjin">@whjin</a>)<br>  on <a href="https://codepen.io">CodePen</a>.</span><br></p></p>
<script async src="https://static.codepen.io/assets/embed/ei.js"></script>

<p>很简单，<code>renderApp</code> 会调用 <code>rendeTitle</code> 和 <code>renderContent</code>，而这两者会把 <code>appState</code> 里面的数据通过原始的 <code>DOM</code> 操作更新到页面上。</p>
<p>这是一个很简单的 <code>App</code>，但是它存在一个重大的隐患，我们渲染数据的时候，使用的是一个共享状态 <code>appState</code>，每个人都可以修改它。</p>
<p><code>renderApp(appState)</code> 之前执行了一大堆函数操作，你根本不知道它们会对 <code>appState</code> 做什么事情，<code>renderApp(appState)</code> 的结果根本没法得到保障。一个可以被不同模块任意修改共享的数据状态就是魔鬼，一旦数据可以任意修改，所有对共享状态的操作都是不可预料的。</p>
<p>但不同的模块（组件）之间确实需要共享数据，这些模块（组件）还可能需要修改这些共享数据，就像上一节的“主题色”状态（<code>themeColor</code>）。这里的矛盾就是：<strong>“模块（组件）之间需要共享数据”，和“数据可能被任意修改导致不可预料的结果”之间的矛盾</strong>。</p>
<p><strong>所有对数据的操作必须通过 <code>dispatch</code> 函数</strong>。它接受一个参数 <code>action</code>，这个 <code>action</code> 是一个普通的 <code>JavaScript</code> 对象，里面必须包含一个 <code>type</code> 字段来声明你到底想干什么。<code>dispatch</code> 在 <code>swtich</code> 里面会识别这个 <code>type</code> 字段，能够识别出来的操作才会执行对 <code>appState</code> 的修改。</p>
<p>上面的 <code>dispatch</code> 它只能识别两种操作，一种是 <code>UPDATE_TITLE_TEXT</code> 它会用 <code>action</code> 的 <code>text</code> 字段去更新 <code>appState.title.text</code>；一种是 <code>UPDATE_TITLE_COLOR</code>，它会用 <code>action</code> 的 <code>color</code> 字段去更新 <code>appState.title.color</code>。可以看到，<code>action</code> 里面除了 <code>type</code> 字段是必须的以外，其他字段都是可以自定义的。</p>
<p>任何的模块如果想要修改 <code>appState.title.text</code>，必须大张旗鼓地调用 <code>dispatch</code>。</p>
<h2 id="redux-store-">Redux-抽离store和监控数据变化</h2>
<h3 id="-store">抽离出 store</h3>
<p>现在我们把它们集中到一个地方，给这个地方起个名字叫做 <code>store</code>，然后构建一个函数 <code>createStore</code>，用来专门生产这种 <code>state</code> 和 <code>dispatch</code> 的集合，这样别的 <code>App</code> 也可以用这种模式了。</p>
<p><code>createStore</code> 接受两个参数，一个是表示应用程序状态的 <code>state</code>；另外一个是 <code>stateChanger</code>，它来描述应用程序状态会根据 <code>action</code> 发生什么变化，其实就是相当于本节开头的 <code>dispatch</code> 代码里面的内容。</p>
<p><code>createStore</code> 会返回一个对象，这个对象包含两个方法 <code>getState</code> 和 <code>dispatch</code>。</p>
<p><code>getState</code> 用于获取 <code>state</code> 数据，其实就是简单地把 <code>state</code> 参数返回。</p>
<p><code>dispatch</code> 用于修改数据，和以前一样会接受 <code>action</code>，然后它会把 <code>state</code> 和 <code>action</code> 一并传给 <code>stateChanger</code>，那么 <code>stateChanger</code> 就可以根据 <code>action</code> 来修改 <code>state</code> 了。</p>
<p>针对每个不同的 <code>App</code>，我们可以给 <code>createStore</code> 传入初始的数据 <code>appState</code>，和一个描述数据变化的函数 <code>stateChanger</code>，然后生成一个 <code>store</code>。需要修改数据的时候通过 <code>store.dispatch</code>，需要获取数据的时候通过 <code>store.getState</code>。</p>
<h3 id="-">监控数据变化</h3>
<p>我们每次通过 <code>dispatch</code> 修改数据的时候，其实只是数据发生了变化，如果我们不手动调用 <code>renderApp</code>，页面上的内容是不会发生变化的。但是我们总不能每次 <code>dispatch</code> 的时候都手动调用一下 <code>renderApp</code>，我们肯定希望数据变化的时候程序能够智能一点地自动重新渲染数据，而不是手动调用。</p>
<p>你说这好办，往 <code>dispatch</code> 里面加 <code>renderApp</code> 就好了，但是这样 <code>createStore</code> 就不够通用了。我们希望用一种通用的方式“<strong>监听</strong>”数据变化，然后重新渲染页面，这里要用到<strong>观察者模式</strong>。</p>
<pre><code>function createStore(state, stateChanger) {
    const listeners = [];
    const subscribe = (listener) =&gt; listeners.push(listener);
    const getState = () =&gt; state;
    const dispatch = (action) =&gt; {
        stateChanger(state, action);
        listeners.forEach((listener) =&gt; listener())
    };
    return {getState, dispatch, subscribe}
}
</code></pre><p>我们在 <code>createStore</code> 里面定义了一个数组 <code>listeners</code>，还有一个新的方法 <code>subscribe</code>，可以通过 <code>store.subscribe(listener)</code> 的方式给 <code>subscribe</code> 传入一个监听函数，这个函数会被 <code>push</code> 到数组当中。</p>
<p>我们修改了 <code>dispatch</code>，每次当它被调用的时候，除了会调用 <code>stateChanger</code> 进行数据的修改，还会遍历 <code>listeners</code> 数组里面的函数，然后一个个地去调用。相当于我们可以通过 <code>subscribe</code> 传入数据变化的监听函数，每当 <code>dispatch</code> 的时候，监听函数就会被调用，这样我们就可以在每当数据变化时候进行重新渲染。</p>
<p>我们只需要 <code>subscribe</code> 一次，后面不管如何 <code>dispatch</code> 进行修改数据，<code>renderApp</code> 函数都会被重新调用，页面就会被重新渲染。这样的订阅模式还有好处就是，以后我们还可以拿同一块数据来渲染别的页面，这时 <code>dispatch</code> 导致的变化也会让每个页面都重新渲染。</p>
<p><p class="codepen" data-height="265" data-theme-id="0" data-default-tab="js,result" data-user="whjin" data-slug-hash="yZjGwj" style="height: 265px; box-sizing: border-box; display: flex; align-items: center; justify-content: center; border: 2px solid black; margin: 1em 0; padding: 1em;" data-pen-title="Redux-抽离 store 和监控数据变化"><br>  <span>See the Pen <a href="https://codepen.io/whjin/pen/yZjGwj/"><br>  Redux-抽离 store 和监控数据变化</a> by whjin (<a href="https://codepen.io/whjin">@whjin</a>)<br>  on <a href="https://codepen.io">CodePen</a>.</span><br></p></p>
<script async src="https://static.codepen.io/assets/embed/ei.js"></script>

<p><strong>总结</strong></p>
<p>现在我们有了一个比较通用的 <code>createStore</code>，它可以产生一种我们新定义的数据类型 <code>store</code>，通过 <code>store.getState</code> 我们获取共享状态，而且我们约定只能通过 <code>store.dispatch</code> 修改共享状态。<code>store</code> 也允许我们通过 <code>store.subscribe</code> 监听数据数据状态被修改了，并且进行后续的例如重新渲染页面的操作。</p>
<h2 id="redux-">Redux-纯函数</h2>
<p><strong>一个函数的返回结果只依赖于它的参数，并且在执行过程里面没有副作用，我们就把这个函数叫做纯函数。</strong></p>
<ol>
<li>函数的返回结果只依赖于它的参数。</li><li>函数执行过程里面没有副作用。</li></ol>
<p>纯函数很严格，也就是说你几乎除了计算数据以外什么都不能干，计算的时候还不能依赖除了函数参数以外的数据。</p>
<h2 id="redux-">Redux-共享结构的对象提高性能</h2>
<p>我们新建一个 <code>appState</code>，新建 <code>appState.title</code>，新建 <code>appState.title.text</code>。</p>
<p><code>appState</code> 和 <code>newAppState</code> 其实是两个不同的对象，因为对象浅复制<code>const appState2={...appState}</code>的缘故，其实它们里面的属性 <code>content</code> 指向的是同一个对象；但是因为 <code>title</code> 被一个新的对象覆盖了，所以它们的 <code>title</code> 属性指向的对象是不同的。</p>
<p>我们每次修改某些数据的时候，都不会碰原来的数据，而是把需要修改数据路径上的对象都 <code>copy</code> 一个出来。</p>
<p>修改数据的时候就把修改路径都复制一遍，但是保持其他内容不变，最后的所有对象具有某些不变共享的结构（例如上面三个对象都共享 <code>content</code> 对象）。大多数情况下我们可以保持 <code>50%</code> 以上的内容具有共享结构，这种操作具有非常优良的特性，我们可以用它来优化上面的渲染性能。</p>
<p><strong>优化性能</strong></p>
<p>我们修改 <code>stateChanger</code>，让它修改数据的时候，并不会直接修改原来的数据 <code>state</code>，而是产生上述的共享结构的对象。</p>
<p>每次需要修改的时候都会产生新的对象，并且返回。而如果没有修改（在 <code>default</code> 语句中）则返回原来的 <code>state</code> 对象。</p>
<p>因为 <code>stateChanger</code> 不会修改原来对象了，而是返回对象，所以我们需要修改一下 <code>createStore</code>。让它用每次 <code>stateChanger(state, action)</code> 的调用结果覆盖原来的 <code>state</code>。</p>
<h2 id="redux-reducer">Redux-reducer</h2>
<p><code>stateChanger</code> 现在既充当了获取初始化数据的功能，也充当了生成更新数据的功能。</p>
<p>如果有传入 <code>state</code> 就生成更新数据，否则就是初始化数据。这样我们可以优化 <code>createStore</code> 成一个参数，因为 <code>state</code> 和 <code>stateChanger</code> 合并到一起了。</p>
<p><code>createStore</code> 内部的 <code>state</code> 不再通过参数传入，而是一个局部变量 <code>let state = null</code>。<code>createStore</code> 的最后会手动调用一次 <code>dispatch({})</code>，<code>dispatch</code> 内部会调用 <code>stateChanger</code>，这时候的 <code>state</code> 是 <code>null</code>，所以这次的 <code>dispatch</code> 其实就是初始化数据了。<code>createStore</code> 内部第一次的 <code>dispatch</code> 导致 <code>state</code> 初始化完成，后续外部的 <code>dispatch</code> 就是修改数据的行为了。</p>
<p>这是一个最终形态的 <code>createStore</code>，它接受的参数叫 <code>reducer</code>，<code>reducer</code> 是一个函数，细心的朋友会发现，它其实是一个纯函数。</p>
<p><strong>reducer</strong></p>
<p><code>createStore</code> 接受一个叫 <code>reducer</code> 的函数作为参数，<strong>这个函数规定是一个纯函数</strong>，它接受两个参数，一个是 <code>state</code>，一个是 <code>action</code>。</p>
<p>如果没有传入 <code>state</code> 或者 <code>state</code> 是 <code>null</code>，那么它就会返回一个初始化的数据。如果有传入 <code>state</code> 的话，就会根据 <code>action</code> 来“<strong>修改</strong>“数据，但其实它没有、也规定不能修改 <code>state</code>，而是要通过上节所说的把修改路径的对象都复制一遍，然后产生一个新的对象返回。如果它不能识别你的 <code>action</code>，它就不会产生新的数据，而是（在 <code>default</code> 内部）把 <code>state</code> 原封不动地返回。</p>
<p><code>reducer</code> 是不允许有副作用的。你不能在里面操作 <code>DOM</code>，也不能发 <code>Ajax</code> 请求，更不能直接修改 <code>state</code>，它要做的仅仅是 —— <strong>初始化和计算新的 <code>state</code></strong>。</p>
<h2 id="redux-redux-">Redux-Redux模式</h2>
<p>你必须通过 <code>dispatch</code> 执行某些允许的修改操作，而且必须大张旗鼓的在 <code>action</code> 里面声明。</p>
<p>这种模式挺好用的，我们就把它抽象出来一个 <code>createStore</code>，它可以产生 <code>store</code>，里面包含 <code>getState</code> 和 <code>dispatch</code> 函数，方便我们使用。</p>
<p>后来发现每次修改数据都需要手动重新渲染非常麻烦，我们希望<strong>自动重新渲染视图</strong>。所以后来加入了<strong>订阅者模式</strong>，可以通过 <code>store.subscribe</code> 订阅数据修改事件，每次数据更新的时候自动重新渲染视图。</p>
<p>我们优化了 <code>stateChanger</code> 为 <code>reducer</code>，定义了 <code>reducer</code> 只能是纯函数，功能就是负责初始化 <code>state</code>，和根据 <code>state</code> 和 <code>action</code> 计算具有共享结构的新的 <code>state</code>。</p>
<blockquote>
<p>原文链接：<a href="http://huziketang.mangojuice.top/books/react/">React.js 小书</a></p>
</blockquote>

</body>
</html>
<!-- This document was created with MarkdownPad, the Markdown editor for Windows (http://markdownpad.com) -->
